五十分钟四题 1A 下班,赛后补了下 E 。
比赛链接:https://codeforces.com/contest/1768 
x ! + ( x − 1 ) ! = ( x + 1 ) ⋅ ( x − 1 ) ! x! + (x - 1)! = (x + 1) \cdot (x - 1)! x ! + ( x − 1 ) ! = ( x + 1 ) ⋅ ( x − 1 ) ! 
当 x = k − 1 x = k - 1 x = k − 1 k ⋅ ( k − 2 ) ! k \cdot (k - 2)! k ⋅ ( k − 2 ) ! k k k 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 #include  <bits/stdc++.h>  using  namespace  std;int  main ()      ios::sync_with_stdio (false );     cin.tie (nullptr );     int  t;     cin >> t;     while  (t--) {         int  k;         cin >> k;         cout << (k - 1 ) << "\n" ;     }     return  0 ; } 
找以 1 开头的最长连续上升子序列,其他数都需要操作后移到尾部,次数即对 k k k 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 #include  <bits/stdc++.h>  using  namespace  std;int  main ()      ios::sync_with_stdio (false );     cin.tie (nullptr );     int  t;     cin >> t;     while  (t--) {         int  n, k;         cin >> n >> k;         vector<int > p (n)  ;         for  (int  i = 0 ; i < n; i++) {             cin >> p[i];         }         int  cnt = n, val = 1 ;         for  (int  i = 0 ; i < n; i++) {             if  (p[i] == val) {                 val++;                 cnt--;             }         }         cout << (cnt + k - 1 ) / k << "\n" ;     }     return  0 ; } 
如果有数出现三次则无解。
否则,将第一次出现的数和第二次出现的数分别赋给 p , q p, q p , q set 的 upper_bound 贪心查找即可。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 #include  <bits/stdc++.h>  using  namespace  std;int  main ()      ios::sync_with_stdio (false );     cin.tie (nullptr );     int  t;     cin >> t;     while  (t--) {         int  n;         cin >> n;         vector<int > a (n) , cnt (n)  ;         for  (int  i = 0 ; i < n; i++) {             cin >> a[i];             --a[i];             cnt[a[i]] += 1 ;         }         bool  ok = true ;         for  (int  i = 0 ; i < n; i++) {             if  (cnt[i] >= 3 ) {                 ok = false ;                 break ;             }         }         if  (not  ok) {             cout << "NO"  << "\n" ;             continue ;         }         vector<int > p (n, -1 ) , q (n, -1 )  ;         vector<bool > vis (n)  ;         for  (int  i = 0 ; i < n; i++) {             if  (not  vis[a[i]]) {                 p[i] = a[i];                 vis[a[i]] = true ;             }         }         for  (int  i = 0 ; i < n; i++) {             if  (vis[a[i]] and  p[i] == -1 ) {                 q[i] = a[i];             }         }         auto  fill_blank = [&](vector<int >& p, vector<int >& q) {             set<int > st;             for  (int  i = 0 ; i < n; i++) {                 st.insert (i);             }             for  (int  i = 0 ; i < n; i++) {                 st.erase (p[i]);             }             for  (int  i = 0 ; i < (int )p.size (); i++) {                 if  (p[i] == -1  and  q[i] != -1 ) {                     auto  it = st.upper_bound (q[i]);                     if  (it == st.begin ()) {                         ok = false ;                         return ;                     }                     --it;                     p[i] = *it;                     st.erase (it);                 }             }         };                  fill_blank (p, q);         fill_blank (q, p);         if  (not  ok) {             cout << "NO"  << "\n" ;         } else  {             cout << "YES"  << "\n" ;             for  (int  i = 0 ; i < n; i++) {                 cout << p[i] + 1  << " \n" [i == n - 1 ];             }             for  (int  i = 0 ; i < n; i++) {                 cout << q[i] + 1  << " \n" [i == n - 1 ];             }         }     }     return  0 ; } 
只有一个逆序对即交换升序排列中的某对相邻值。
找出 p p p k k k k − 1 k - 1 k − 1 
若某个循环节里存在相邻值,则最后排为升序时可以减少一次操作来使这两个相邻值形成逆序。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 #include  <bits/stdc++.h>  using  namespace  std;int  main ()      ios::sync_with_stdio (false );     cin.tie (nullptr );     int  t;     cin >> t;     while  (t--) {         int  n;         cin >> n;         vector<int > p (n)  ;         for  (int  i = 0 ; i < n; i++) {             cin >> p[i];             --p[i];         }         int  ans = 0 ;         bool  ok = false ;         vector<bool > vis (n)  ;         for  (int  i = 0 ; i < n; i++) {             if  (vis[i]) {                 continue ;             }             vector<int > v;             int  j = i, res = -1 ;             while  (not  vis[j]) {                 vis[j] = true ;                 v.push_back (p[j]);                 j = p[j];                 res += 1 ;             }             sort (v.begin (), v.end ());             for  (int  j = 0 ; j + 1  < (int )v.size (); j++) {                 if  (v[j] + 1  == v[j + 1 ]) {                     ok = true ;                 }             }             ans += res;         }         cout << (ans + (ok ? -1  : 1 )) << "\n" ;     }     return  0 ; } 
当 [ 1 , n ] [1, n] [ 1 , n ] n n n [ 2 n + 1 , 3 n ] [2n + 1, 3n] [ 2 n + 1 , 3 n ] n n n 3 3 3  
当 [ 1 , n ] [1, n] [ 1 , n ] 2 n 2n 2 n [ 2 n + 1 , 3 n ] [2n + 1, 3n] [ 2 n + 1 , 3 n ] 2 n 2n 2 n 2 2 2  
当前 n n n n n n 1 1 1  
当 3 n 3n 3 n  
 
由于不好单独计算某一操作次数的排列数,可以用容斥原理从低到高依次计算。
操作 0 0 0 3 n 3n 3 n 1 1 1  
操作 ≤ 1 \le 1 ≤ 1 
前 n n n 2 n ! 2n! 2 n !  
后 n n n 2 n ! 2n! 2 n !  
二者交集,即前后 n n n [ n + 1 , 2 n ] [n + 1, 2n] [ n + 1 , 2 n ] n n n n ! n! n !  
所以共有 2 ⋅ 2 n ! − n ! 2 \cdot 2n! - n! 2 ⋅ 2 n ! − n !  
 
 
操作 ≤ 2 \le 2 ≤ 2 
[ 1 , n ] [1, n] [ 1 , n ] 2 n 2n 2 n C 2 n n ⋅ n ! ⋅ 2 n ! C_{2n}^{n} \cdot n! \cdot 2n! C 2 n n  ⋅ n ! ⋅ 2 n ! [ 2 n + 1 , 3 n ] [2n + 1, 3n] [ 2 n + 1 , 3 n ] 2 n 2n 2 n C 2 n n ⋅ n ! ⋅ 2 n ! C_{2n}^{n} \cdot n! \cdot 2n! C 2 n n  ⋅ n ! ⋅ 2 n ! 二者交集,即 [ 1 , n ] [1, n] [ 1 , n ] 2 n 2n 2 n [ 2 n + 1 , 3 n ] [2n + 1, 3n] [ 2 n + 1 , 3 n ] 2 n 2n 2 n 
设 [ 1 , n ] [1, n] [ 1 , n ] n n n i i i n n n n − i n - i n − i C n i ⋅ C n n − i ⋅ n ! C_{n}^{i} \cdot C_{n}^{n - i} \cdot n! C n i  ⋅ C n n − i  ⋅ n !  
此时 [ 2 n + 1 , 3 n ] [2n + 1, 3n] [ 2 n + 1 , 3 n ] n n n n − i n - i n − i 2 n − ( n − i ) 2n - (n - i) 2 n − ( n − i ) i i i C 2 n − ( n − i ) i ⋅ n ! C_{2n - (n - i)}^{i} \cdot n! C 2 n − ( n − i ) i  ⋅ n !  
最后 [ 2 n + 1 , 3 n ] [2n + 1, 3n] [ 2 n + 1 , 3 n ] 2 n 2n 2 n n n n n ! n! n !  
所以共有 2 ⋅ C 2 n n ⋅ n ! ⋅ 2 n ! − ∑ i = 0 n ( C n i ⋅ C n n − i ⋅ n ! ) ⋅ ( C 2 n − ( n − i ) i ⋅ n ! ) ⋅ ( n ! ) 2 \cdot C_{2n}^{n} \cdot n! \cdot 2n! - \sum \limits _{i = 0} ^{n} (C_{n}^{i} \cdot C_{n}^{n - i} \cdot n!) \cdot (C_{2n - (n - i)}^{i} \cdot n!) \cdot (n!) 2 ⋅ C 2 n n  ⋅ n ! ⋅ 2 n ! − i = 0 ∑ n  ( C n i  ⋅ C n n − i  ⋅ n ! ) ⋅ ( C 2 n − ( n − i ) i  ⋅ n ! ) ⋅ ( n ! )  
 
 
 
 
操作 ≤ 3 \le 3 ≤ 3  
 
依次相减得出操作次数等于 i i i c n t i cnt_i c n t i  ∑ i = 1 3 i ⋅ c n t i \sum \limits _{i = 1} ^{3} i \cdot cnt_i i = 1 ∑ 3  i ⋅ c n t i  
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 #include  <bits/stdc++.h>  using  namespace  std;int  MOD = 998244353 ;int  norm (int  x)  if  (x < 0 ) { x += MOD; } if  (x >= MOD) { x -= MOD; } return  x; }template <class  T> T binpow (T a, int  b)  1 ; for  (; b; b /= 2 , a *= a) { if  (b % 2 ) { res *= a; } } return  res; }struct  Z  {    int  x;     Z (int  x = 0 ) : x (norm (x)) {}     int  val ()  const  return  x; }     Z operator -() const  { return  Z (norm (MOD - x)); }     Z inv ()  const   { assert (x != 0 ); return  binpow (*this , MOD - 2 ); }     Z &operator *=(const  Z &rhs) { x = 1LL  * x * rhs.x % MOD; return  *this ; }     Z &operator +=(const  Z &rhs) { x = norm (x + rhs.x); return  *this ; }     Z &operator -=(const  Z &rhs) { x = norm (x - rhs.x); return  *this ; }     Z &operator /=(const  Z &rhs) { return  *this  *= rhs.inv (); }     friend  Z operator *(const  Z &lhs, const  Z &rhs) { Z res = lhs; res *= rhs; return  res; }     friend  Z operator +(const  Z &lhs, const  Z &rhs) { Z res = lhs; res += rhs; return  res; }     friend  Z operator -(const  Z &lhs, const  Z &rhs) { Z res = lhs; res -= rhs; return  res; }     friend  Z operator /(const  Z &lhs, const  Z &rhs) { Z res = lhs; res /= rhs; return  res; } }; struct  Combination  {    vector<Z> fac, inv;     Combination (int  n) : fac (n), inv (n) {         fac[0 ] = 1 ;         for  (int  i = 1 ; i < n; i++) fac[i] = fac[i - 1 ] * i;         inv[n - 1 ] = fac[n - 1 ].inv ();         for  (int  i = n - 2 ; i >= 0 ; i--) inv[i] = inv[i + 1 ] * (i + 1 );     }     Z C (int  n, int  m)  {         if (m < 0  or  m > n) return  0 ;         return  fac[n] * inv[m] * inv[n - m];     }     Z A (int  n, int  m)  {         if (m < 0  or  m > n) return  0 ;         return  fac[n] * inv[n - m];     } }; int  main ()      ios::sync_with_stdio (false );     cin.tie (nullptr );     int  n;     cin >> n >> MOD;     Combination C (3  * n + 1 )  ;     Z cnt0 = 1 ;     Z cnt1 = Z (2 ) * C.fac[2  * n] - C.fac[n] - cnt0;     Z cnt2 = Z (2 ) * C.C (2  * n, n) * C.fac[n] * C.fac[2  * n] - cnt1 - cnt0;     for  (int  i = 0 ; i <= n; i++) {         cnt2 -= C.C (n, i) * C.C (n, n - i) * C.C (2  * n - (n - i), i) * C.fac[n] * C.fac[n] * C.fac[n];     }     Z cnt3 = C.fac[3  * n] - cnt2 - cnt1 - cnt0;     cout << (cnt1 + 2  * cnt2 + 3  * cnt3).val () << "\n" ;     return  0 ; }